home *** CD-ROM | disk | FTP | other *** search
/ CU Amiga Super CD-ROM 15 / CU Amiga Magazine's Super CD-ROM 15 (1997)(EMAP Images)(GB)[!][issue 1997-10].iso / CUCD / Graphics / Ghostscript / source / gdevpcx.c < prev    next >
C/C++ Source or Header  |  1997-03-26  |  14KB  |  462 lines

  1. /* Copyright (C) 1992, 1995, 1996, 1997 Aladdin Enterprises.  All rights reserved.
  2.   
  3.   This file is part of Aladdin Ghostscript.
  4.   
  5.   Aladdin Ghostscript is distributed with NO WARRANTY OF ANY KIND.  No author
  6.   or distributor accepts any responsibility for the consequences of using it,
  7.   or for whether it serves any particular purpose or works at all, unless he
  8.   or she says so in writing.  Refer to the Aladdin Ghostscript Free Public
  9.   License (the "License") for full details.
  10.   
  11.   Every copy of Aladdin Ghostscript must include a copy of the License,
  12.   normally in a plain ASCII text file named PUBLIC.  The License grants you
  13.   the right to copy, modify and redistribute Aladdin Ghostscript, but only
  14.   under certain conditions described in the License.  Among other things, the
  15.   License requires that the copyright notice and this notice be preserved on
  16.   all copies.
  17. */
  18.  
  19. /* gdevpcx.c */
  20. /* PCX file format drivers */
  21. #include "gdevprn.h"
  22. #include "gdevpccm.h"
  23. #include "gxlum.h"
  24.  
  25. /* Thanks to Phil Conrad for donating the original version */
  26. /* of these drivers to Aladdin Enterprises. */
  27.  
  28. /* ------ The device descriptors ------ */
  29.  
  30. /*
  31.  * Default X and Y resolution.
  32.  */
  33. #define X_DPI 72
  34. #define Y_DPI 72
  35.  
  36. /* Monochrome. */
  37.  
  38. private dev_proc_print_page(pcxmono_print_page);
  39.  
  40. /* Use the default RGB->color map, so we get black=0, white=1. */
  41. private const gx_device_procs pcxmono_procs =
  42.   prn_color_procs(gdev_prn_open, gdev_prn_output_page, gdev_prn_close,
  43.     gx_default_map_rgb_color, gx_default_map_color_rgb);
  44. gx_device_printer far_data gs_pcxmono_device =
  45.   prn_device(pcxmono_procs, "pcxmono",
  46.          DEFAULT_WIDTH_10THS, DEFAULT_HEIGHT_10THS,
  47.          X_DPI, Y_DPI,
  48.          0,0,0,0,            /* margins */
  49.          1, pcxmono_print_page);
  50.  
  51. /* Chunky 8-bit gray scale. */
  52.  
  53. private dev_proc_print_page(pcx256_print_page);
  54.  
  55. private const gx_device_procs pcxgray_procs =
  56.   prn_color_procs(gdev_prn_open, gdev_prn_output_page, gdev_prn_close,
  57.     gx_default_gray_map_rgb_color, gx_default_gray_map_color_rgb);
  58. gx_device_printer far_data gs_pcxgray_device =
  59. {  prn_device_body(gx_device_printer, pcxgray_procs, "pcxgray",
  60.            DEFAULT_WIDTH_10THS, DEFAULT_HEIGHT_10THS,
  61.            X_DPI, Y_DPI,
  62.            0,0,0,0,            /* margins */
  63.            1,8,255,0,256,0, pcx256_print_page)
  64. };
  65.  
  66. /* 4-bit planar (EGA/VGA-style) color. */
  67.  
  68. private dev_proc_print_page(pcx16_print_page);
  69.  
  70. private const gx_device_procs pcx16_procs =
  71.   prn_color_procs(gdev_prn_open, gdev_prn_output_page, gdev_prn_close,
  72.     pc_4bit_map_rgb_color, pc_4bit_map_color_rgb);
  73. gx_device_printer far_data gs_pcx16_device =
  74. {  prn_device_body(gx_device_printer, pcx16_procs, "pcx16",
  75.            DEFAULT_WIDTH_10THS, DEFAULT_HEIGHT_10THS,
  76.            X_DPI, Y_DPI,
  77.            0,0,0,0,            /* margins */
  78.            3,4,3,2,4,3, pcx16_print_page)
  79. };
  80.  
  81. /* Chunky 8-bit (SuperVGA-style) color. */
  82. /* (Uses a fixed palette of 3,3,2 bits.) */
  83.  
  84. private const gx_device_procs pcx256_procs =
  85.   prn_color_procs(gdev_prn_open, gdev_prn_output_page, gdev_prn_close,
  86.     pc_8bit_map_rgb_color, pc_8bit_map_color_rgb);
  87. gx_device_printer far_data gs_pcx256_device =
  88. {  prn_device_body(gx_device_printer, pcx256_procs, "pcx256",
  89.            DEFAULT_WIDTH_10THS, DEFAULT_HEIGHT_10THS,
  90.            X_DPI, Y_DPI,
  91.            0,0,0,0,            /* margins */
  92.            3,8,6,6,7,7, pcx256_print_page)
  93. };
  94.  
  95. /* 24-bit color, 3 8-bit planes. */
  96.  
  97. private dev_proc_print_page(pcx24b_print_page);
  98.  
  99. private const gx_device_procs pcx24b_procs =
  100.   prn_color_procs(gdev_prn_open, gdev_prn_output_page, gdev_prn_close,
  101.     gx_default_rgb_map_rgb_color, gx_default_rgb_map_color_rgb);
  102. gx_device_printer far_data gs_pcx24b_device =
  103.   prn_device(pcx24b_procs, "pcx24b",
  104.          DEFAULT_WIDTH_10THS, DEFAULT_HEIGHT_10THS,
  105.          X_DPI, Y_DPI,
  106.          0,0,0,0,            /* margins */
  107.          24, pcx24b_print_page);
  108.  
  109. /* 4-bit chunky CMYK color. */
  110.  
  111. private dev_proc_print_page(pcxcmyk_print_page);
  112.  
  113. private gx_color_index
  114. pcx_cmyk_map_cmyk_color(gx_device *dev,
  115.   gx_color_value c, gx_color_value m, gx_color_value y, gx_color_value k)
  116. {
  117. #define cv_bit(v) ((v) >> (gx_color_value_bits - 1))
  118.     return (gx_color_index)
  119.       (cv_bit(k) + (cv_bit(y) << 1) + (cv_bit(m) << 2) + (cv_bit(c) << 3));
  120. #undef cv_bit
  121. }
  122. private int
  123. pcx_cmyk_map_color_rgb(gx_device *dev, gx_color_index color,
  124.   gx_color_value prgb[3])
  125. {    if ( color & 1 )
  126.       prgb[0] = prgb[1] = prgb[2] = 0;
  127.     else
  128.       prgb[0] = (color & 8 ? 0 : gx_max_color_value),
  129.       prgb[1] = (color & 4 ? 0 : gx_max_color_value),
  130.       prgb[2] = (color & 2 ? 0 : gx_max_color_value);
  131.     return 0;
  132. }
  133. private const gx_device_procs pcxcmyk_procs = {
  134.     gdev_prn_open,
  135.     NULL,    /* get_initial_matrix */
  136.     NULL,    /* sync_output */
  137.     gdev_prn_output_page,
  138.     gdev_prn_close,
  139.     NULL,    /* map_rgb_color */
  140.     pcx_cmyk_map_color_rgb,
  141.     NULL,    /* fill_rectangle */
  142.     NULL,    /* tile_rectangle */
  143.     NULL,    /* copy_mono */
  144.     NULL,    /* copy_color */
  145.     NULL,    /* draw_line */
  146.     NULL,    /* get_bits */
  147.     gdev_prn_get_params,
  148.     gdev_prn_put_params,
  149.     pcx_cmyk_map_cmyk_color,    /* map_cmyk_color */
  150.     NULL,    /* get_xfont_procs */
  151.     NULL,    /* get_xfont_device */
  152.     NULL,    /* map_rgb_alpha_color */
  153.     gx_page_device_get_page_device
  154. };
  155. gx_device_printer far_data gs_pcxcmyk_device =
  156. {  prn_device_body(gx_device_printer, pcxcmyk_procs, "pcxcmyk",
  157.            DEFAULT_WIDTH_10THS, DEFAULT_HEIGHT_10THS,
  158.            X_DPI, Y_DPI,
  159.            0,0,0,0,            /* margins */
  160.            4,4,1,1,2,2, pcxcmyk_print_page)
  161. };
  162.  
  163. /* ------ Private definitions ------ */
  164.  
  165. /* All two-byte quantities are stored LSB-first! */
  166. #if arch_is_big_endian
  167. #  define assign_ushort(a,v) a = ((v) >> 8) + ((v) << 8)
  168. #else
  169. #  define assign_ushort(a,v) a = (v)
  170. #endif
  171.  
  172. typedef struct pcx_header_s {
  173.     byte     manuf;        /* always 0x0a */
  174.     byte    version;    /* version info = 0,2,3,5 */
  175.     byte    encoding;    /* 1=RLE */
  176.     byte    bpp;        /* bits per pixel per plane */
  177.     ushort    x1;        /* X of upper left corner */
  178.     ushort    y1;        /* Y of upper left corner */
  179.     ushort    x2;        /* x1 + width - 1 */
  180.     ushort    y2;        /* y1 + height - 1 */
  181.     ushort    hres;        /* horz. resolution (dots per inch) */
  182.     ushort    vres;        /* vert. resolution (dots per inch) */
  183.     byte    palette[16*3];    /* color palette */
  184.     byte    reserved;
  185.     byte    nplanes;    /* number of color planes */
  186.     ushort    bpl;        /* number of bytes per line (uncompressed) */
  187.     ushort    palinfo;    /* palette info 1=color, 2=grey */
  188.     byte    xtra[58];    /* fill out header to 128 bytes */
  189. } pcx_header;
  190. /* Define the prototype header. */
  191. private const pcx_header far_data pcx_header_prototype = {
  192.     10,            /* manuf */
  193.     5,            /* version */
  194.     1,            /* encoding */
  195.     0,            /* bpp (variable) */
  196.     00, 00,            /* x1, y1 */
  197.     00, 00,            /* x2, y2 (variable) */
  198.     00, 00,            /* hres, vres (variable) */
  199.       {0,0,0, 0,0,0, 0,0,0, 0,0,0,    /* palette (variable) */
  200.        0,0,0, 0,0,0, 0,0,0, 0,0,0,
  201.        0,0,0, 0,0,0, 0,0,0, 0,0,0,
  202.        0,0,0, 0,0,0, 0,0,0, 0,0,0},
  203.     0,            /* reserved */
  204.     0,            /* nplanes (variable) */
  205.     00,            /* bpl (variable) */
  206.     00,            /* palinfo (variable) */
  207.       {            0,0, 0,0,0,0,0,0,0,0,    /* xtra */
  208.        0,0,0,0,0,0,0,0, 0,0,0,0,0,0,0,0,
  209.        0,0,0,0,0,0,0,0, 0,0,0,0,0,0,0,0,
  210.        0,0,0,0,0,0,0,0, 0,0,0,0,0,0,0,0}
  211. };
  212.  
  213. /* 
  214. ** version info for PCX is as follows 
  215. **
  216. ** 0 == 2.5
  217. ** 2 == 2.8 w/palette info
  218. ** 3 == 2.8 without palette info
  219. ** 5 == 3.0 (includes palette)
  220. **
  221. */
  222.  
  223. /*
  224.  * Define the DCX header.  We don't actually use this yet.
  225.  * All quantities are stored little-endian!
  226.     bytes 0-3: ID = 987654321
  227.     bytes 4-7: file offset of page 1
  228.     [... up to 1023 entries ...]
  229.     bytes N-N+3: 0 to mark end of page list
  230.  * This is followed by the pages in order, each of which is a PCX file.
  231.  */
  232. #define dcx_magic 987654321
  233. #define dcx_max_pages 1023
  234.  
  235. /* Forward declarations */
  236. private void pcx_write_rle(P4(const byte *, const byte *, int, FILE *));
  237. private int pcx_write_page(P4(gx_device_printer *, FILE *, pcx_header _ss *, bool));
  238.  
  239. /* Write a monochrome PCX page. */
  240. private int
  241. pcxmono_print_page(gx_device_printer *pdev, FILE *file)
  242. {    pcx_header header;
  243.  
  244.     header = pcx_header_prototype;
  245.     header.version = 2;
  246.     header.bpp = 1;
  247.     header.nplanes = 1;
  248.     /* Set the first two entries of the short palette. */
  249.     memcpy((byte *)header.palette, "\000\000\000\377\377\377", 6);
  250.     return pcx_write_page(pdev, file, &header, false);
  251. }
  252.  
  253. /* Write an "old" PCX page. */
  254. static const byte pcx_ega_palette[16*3] = {
  255.   0x00,0x00,0x00,  0x00,0x00,0xaa,  0x00,0xaa,0x00,  0x00,0xaa,0xaa,
  256.   0xaa,0x00,0x00,  0xaa,0x00,0xaa,  0xaa,0xaa,0x00,  0xaa,0xaa,0xaa,
  257.   0x55,0x55,0x55,  0x55,0x55,0xff,  0x55,0xff,0x55,  0x55,0xff,0xff,
  258.   0xff,0x55,0x55,  0xff,0x55,0xff,  0xff,0xff,0x55,  0xff,0xff,0xff
  259. };
  260. private int
  261. pcx16_print_page(gx_device_printer *pdev, FILE *file)
  262. {    pcx_header header;
  263.  
  264.     header = pcx_header_prototype;
  265.     header.version = 2;
  266.     header.bpp = 1;
  267.     header.nplanes = 4;
  268.     /* Fill the EGA palette appropriately. */
  269.     memcpy((byte *)header.palette, pcx_ega_palette,
  270.            sizeof(pcx_ega_palette));
  271.     return pcx_write_page(pdev, file, &header, true);
  272. }
  273.  
  274. /* Write a "new" PCX page. */
  275. private int
  276. pcx256_print_page(gx_device_printer *pdev, FILE *file)
  277. {    pcx_header header;
  278.     int code;
  279.  
  280.     header = pcx_header_prototype;
  281.     header.bpp = 8;
  282.     header.nplanes = 1;
  283.     code = pcx_write_page(pdev, file, &header, false);
  284.     if ( code >= 0 )
  285.     {    /* Write out the palette. */
  286.         fputc(0x0c, file);
  287.         code = pc_write_palette((gx_device *)pdev, 256, file);
  288.     }
  289.     return code;
  290. }
  291.  
  292. /* Write a 24-bit color PCX page. */
  293. private int
  294. pcx24b_print_page(gx_device_printer *pdev, FILE *file)
  295. {    pcx_header header;
  296.  
  297.     header = pcx_header_prototype;
  298.     header.bpp = 8;
  299.     header.nplanes = 3;
  300.     return pcx_write_page(pdev, file, &header, true);
  301. }
  302.  
  303. /* Write a 4-bit chunky CMYK color PCX page. */
  304. static const byte pcx_cmyk_palette[16*3] = {
  305.   0xff,0xff,0xff,  0x00,0x00,0x00,  0xff,0xff,0x00,  0x0f,0x0f,0x00,
  306.   0xff,0x00,0xff,  0x0f,0x00,0x0f,  0xff,0x00,0x00,  0x0f,0x00,0x00,
  307.   0x00,0xff,0xff,  0x00,0x0f,0x0f,  0x00,0xff,0x00,  0x00,0x0f,0x00,
  308.   0x00,0x00,0xff,  0x00,0x00,0x0f,  0x1f,0x1f,0x1f,  0x0f,0x0f,0x0f,
  309. };
  310. private int
  311. pcxcmyk_print_page(gx_device_printer *pdev, FILE *file)
  312. {    pcx_header header;
  313.  
  314.     header = pcx_header_prototype;
  315.     header.version = 2;
  316.     header.bpp = 4;
  317.     header.nplanes = 1;
  318.     /* Fill the palette appropriately. */
  319.     memcpy((byte *)header.palette, pcx_cmyk_palette,
  320.            sizeof(pcx_cmyk_palette));
  321.     return pcx_write_page(pdev, file, &header, false);
  322. }
  323.  
  324. /* Write out a page in PCX format. */
  325. /* This routine is used for all formats. */
  326. /* The caller has set header->bpp, nplanes, and palette. */
  327. private int
  328. pcx_write_page(gx_device_printer *pdev, FILE *file, pcx_header _ss *phdr,
  329.   bool planar)
  330. {    int raster = gdev_prn_raster(pdev);
  331.     uint rsize = round_up((pdev->width * phdr->bpp + 7) >> 3, 2);    /* PCX format requires even */
  332.     int height = pdev->height;
  333.     int depth = pdev->color_info.depth;
  334.     uint lsize = raster + rsize;
  335.     byte *line = (byte *)gs_malloc(lsize, 1, "pcx file buffer");
  336.     byte *plane = line + raster;
  337.     int y;
  338.     int code = 0;            /* return code */
  339.     if ( line == 0 )        /* can't allocate line buffer */
  340.       return_error(gs_error_VMerror);
  341.  
  342.     /* Fill in the variable entries in the header struct. */
  343.  
  344.     assign_ushort(phdr->x2, pdev->width-1);
  345.     assign_ushort(phdr->y2, height-1);
  346.     assign_ushort(phdr->hres, (int)pdev->x_pixels_per_inch);
  347.     assign_ushort(phdr->vres, (int)pdev->y_pixels_per_inch);
  348.     assign_ushort(phdr->bpl, (planar || depth == 1 ? rsize :
  349.                   raster + (raster & 1)));
  350.     assign_ushort(phdr->palinfo, (depth > 1 ? 1 : 2));
  351.  
  352.     /* Write the header. */
  353.  
  354.     if ( fwrite((const char *)phdr, 1, 128, file) < 128 )
  355.     {    code = gs_error_ioerror;
  356.         goto pcx_done;
  357.     }
  358.  
  359.     /* Write the contents of the image. */
  360.     for ( y = 0; y < height; y++ )
  361.     {    byte *row;
  362.         byte *end;
  363.  
  364.         code = gdev_prn_get_bits(pdev, y, line, &row);
  365.         if ( code < 0 ) break;
  366.         end = row + raster;
  367.         if ( !planar )
  368.         {    /* Just write the bits. */
  369.             if ( raster & 1 )
  370.             {    /* Round to even, with predictable padding. */
  371.                 *end = end[-1];
  372.                 ++end;
  373.             }
  374.             pcx_write_rle(row, end, 1, file);
  375.         }
  376.         else
  377.           switch ( depth )
  378.         {
  379.  
  380.         case 4:
  381.         {    byte *pend = plane + rsize;
  382.             int shift;
  383.  
  384.             for ( shift = 0; shift < 4; shift++ )
  385.             {    register byte *from, *to;
  386.                 register int bright = 1 << shift;
  387.                 register int bleft = bright << 4;
  388.  
  389.                 for ( from = row, to = plane;
  390.                       from < end; from += 4
  391.                     )
  392.                 {    *to++ =
  393.                       (from[0] & bleft ? 0x80 : 0) |
  394.                       (from[0] & bright ? 0x40 : 0) |
  395.                       (from[1] & bleft ? 0x20 : 0) |
  396.                       (from[1] & bright ? 0x10 : 0) |
  397.                       (from[2] & bleft ? 0x08 : 0) |
  398.                       (from[2] & bright ? 0x04 : 0) |
  399.                       (from[3] & bleft ? 0x02 : 0) |
  400.                       (from[3] & bright ? 0x01 : 0);
  401.                 }
  402.                 /* We might be one byte short of rsize. */
  403.                 if ( to < pend )
  404.                   *to = to[-1];
  405.                 pcx_write_rle(plane, pend, 1, file);
  406.             }
  407.         }
  408.             break;
  409.  
  410.         case 24:
  411.         {    int pnum;
  412.             for ( pnum = 0; pnum < 3; ++pnum )
  413.               { pcx_write_rle(row + pnum, row + raster, 3, file);
  414.                 if ( pdev->width & 1 )
  415.                   fputc(0, file);        /* pad to even */
  416.               }
  417.         }
  418.             break;
  419.  
  420.         default:
  421.               code = gs_note_error(gs_error_rangecheck);
  422.               goto pcx_done;
  423.  
  424.         }
  425.     }
  426.  
  427. pcx_done:
  428.     gs_free((char *)line, lsize, 1, "pcx file buffer");
  429.  
  430.     return code;
  431. }
  432.  
  433. /* ------ Internal routines ------ */
  434.  
  435. /* Write one line in PCX run-length-encoded format. */
  436. private void
  437. pcx_write_rle(const byte *from, const byte *end, int step, FILE *file)
  438. {    int max_run = step * 63;
  439.     while ( from < end )
  440.     {    byte data = *from;
  441.         from += step;
  442.         if ( data != *from || from == end )
  443.           {    if ( data >= 0xc0 )
  444.               putc(0xc1, file);
  445.           }
  446.         else
  447.           {    const byte *start = from;
  448.             while ( (from < end) && (*from == data) )
  449.               from += step;
  450.             /* Now (from - start) / step + 1 is the run length. */
  451.             while ( from - start >= max_run )
  452.               { putc(0xff, file);
  453.                 putc(data, file);
  454.                 start += max_run;
  455.               }
  456.             if ( from > start || data >= 0xc0 )
  457.               putc((from - start) / step + 0xc1, file);
  458.           }
  459.         putc(data, file);
  460.     }
  461. }
  462.